home *** CD-ROM | disk | FTP | other *** search
- /* dl_next.xs
- *
- * Platform: NeXT NS 3.2
- * Author: Anno Siegel (siegel@zrz.TU-Berlin.DE)
- * Based on: dl_dlopen.xs by Paul Marquess
- * Created: Aug 15th, 1994
- *
- */
-
- /*
- And Gandalf said: 'Many folk like to know beforehand what is to
- be set on the table; but those who have laboured to prepare the
- feast like to keep their secret; for wonder makes the words of
- praise louder.'
- */
-
- /* Porting notes:
-
- dl_next.xs is itself a port from dl_dlopen.xs by Paul Marquess. It
- should not be used as a base for further ports though it may be used
- as an example for how dl_dlopen.xs can be ported to other platforms.
-
- The method used here is just to supply the sun style dlopen etc.
- functions in terms of NeXTs rld_*. The xs code proper is unchanged
- from Paul's original.
-
- The port could use some streamlining. For one, error handling could
- be simplified.
-
- Anno Siegel
-
- */
-
- #include "EXTERN.h"
- #include "perl.h"
- #include "XSUB.h"
-
- #include "dlutils.c" /* SaveError() etc */
-
- static char *dl_last_error = (char *) 0;
-
- static char *dlerror()
- {
- return dl_last_error;
- }
-
- int dlclose(handle) /* stub only */
- void *handle;
- {
- return 0;
- }
-
- #if NS_TARGET_MAJOR >= 4
- #import <mach-o/dyld.h>
-
- enum dyldErrorSource
- {
- OFImage,
- };
-
- static void TranslateError
- (const char *path, enum dyldErrorSource type, int number)
- {
- char errorBuffer[128];
- unsigned int index;
- static char *OFIErrorStrings[] =
- {
- "%s(%d): Object Image Load Failure\n",
- "%s(%d): Object Image Load Success\n",
- "%s(%d): Not an recognisable object file\n",
- "%s(%d): No valid architecture\n",
- "%s(%d): Object image has an invalid format\n",
- "%s(%d): Invalid access (permissions?)\n",
- "%s(%d): Unknown error code from NSCreateObjectFileImageFromFile\n",
- };
- #define NUM_OFI_ERRORS (sizeof(OFIErrorStrings) / sizeof(OFIErrorStrings[0]))
-
- if ( dl_last_error ) {
- safefree(dl_last_error);
- }
- switch (type)
- {
- case OFImage:
- index = number;
- if (index > NUM_OFI_ERRORS - 1)
- index = NUM_OFI_ERRORS - 1;
- sprintf(errorBuffer, OFIErrorStrings[index], path, number);
- break;
-
- default:
- sprintf(errorBuffer, "%s(%d): Totally unknown error type %d\n",
- path, number, type);
- break;
- }
- dl_last_error = safemalloc(strlen(errorBuffer)+1);
- strcpy(dl_last_error, errorBuffer);
- }
-
- static char *dlopen(char *path, int mode /* mode is ignored */)
- {
- int dyld_result;
- NSObjectFileImage ofile;
- NSModule handle = NULL;
-
- dyld_result = NSCreateObjectFileImageFromFile(path, &ofile);
- if (dyld_result != NSObjectFileImageSuccess)
- TranslateError(path, OFImage, dyld_result);
- else
- {
- // NSLinkModule will cause the run to abort on any link error's
- // not very friendly but the error recovery functionality is limited.
- handle = NSLinkModule(ofile, path, TRUE);
- }
-
- return handle;
- }
-
- void *
- dlsym(handle, symbol)
- void *handle;
- char *symbol;
- {
- void *addr;
-
- if (NSIsSymbolNameDefined(symbol))
- addr = NSAddressOfSymbol(NSLookupAndBindSymbol(symbol));
- else
- addr = NULL;
-
- return addr;
- }
-
- #else /* NS_TARGET_MAJOR <= 3 */
-
- #include <streams/streams.h>
- #include <mach-o/rld.h>
-
- static NXStream *OpenError(void)
- {
- return NXOpenMemory( (char *) 0, 0, NX_WRITEONLY);
- }
-
- static void TransferError(NXStream *s)
- {
- char *buffer;
- int len, maxlen;
-
- if ( dl_last_error ) {
- safefree(dl_last_error);
- }
- NXGetMemoryBuffer(s, &buffer, &len, &maxlen);
- dl_last_error = safemalloc(len);
- strcpy(dl_last_error, buffer);
- }
-
- static void CloseError(NXStream *s)
- {
- if ( s ) {
- NXCloseMemory(s, NX_FREEBUFFER);
- }
- }
-
- static char *dlopen(char *path, int mode /* mode is ignored */)
- {
- int rld_success;
- NXStream *nxerr = OpenError();
- AV * av_resolve;
- I32 i, psize;
- char *result;
- char **p;
-
- av_resolve = GvAVn(gv_fetchpv(
- "DynaLoader::dl_resolve_using", FALSE, SVt_PVAV));
- psize = AvFILL(av_resolve) + 3;
- p = (char **) safemalloc(psize * sizeof(char*));
- p[0] = path;
- for(i=1; i<psize-1; i++) {
- p[i] = SvPVx(*av_fetch(av_resolve, i-1, TRUE), na);
- }
- p[psize-1] = 0;
- rld_success = rld_load(nxerr, (struct mach_header **)0, p,
- (const char *) 0);
- safefree((char*) p);
- if (rld_success) {
- result = path;
- } else {
- TransferError(nxerr);
- result = (char*) 0;
- }
- CloseError(nxerr);
- return result;
- }
-
- void *
- dlsym(handle, symbol)
- void *handle;
- char *symbol;
- {
- NXStream *nxerr = OpenError();
- unsigned long symref = 0;
-
- if (!rld_lookup(nxerr, symbol, &symref)) {
- TransferError(nxerr);
- }
- CloseError(nxerr);
- return (void*) symref;
- }
-
- #endif /* NS_TARGET_MAJOR >= 4 */
-
- /* ----- code from dl_dlopen.xs below here ----- */
-
-
- static void
- dl_private_init()
- {
- (void)dl_generic_private_init();
- }
-
- MODULE = DynaLoader PACKAGE = DynaLoader
-
- BOOT:
- (void)dl_private_init();
-
-
-
- void *
- dl_load_file(filename)
- char * filename
- CODE:
- int mode = 1;
- DLDEBUG(1,fprintf(stderr,"dl_load_file(%s):\n", filename));
- RETVAL = dlopen(filename, mode) ;
- DLDEBUG(2,fprintf(stderr," libref=%x\n", RETVAL));
- ST(0) = sv_newmortal() ;
- if (RETVAL == NULL)
- SaveError("%s",dlerror()) ;
- else
- sv_setiv( ST(0), (IV)RETVAL);
-
-
- void *
- dl_find_symbol(libhandle, symbolname)
- void * libhandle
- char * symbolname
- CODE:
- char symbolname_buf[1024];
- symbolname = dl_add_underscore(symbolname, symbolname_buf);
- DLDEBUG(2,fprintf(stderr,"dl_find_symbol(handle=%x, symbol=%s)\n",
- libhandle, symbolname));
- RETVAL = dlsym(libhandle, symbolname);
- DLDEBUG(2,fprintf(stderr," symbolref = %x\n", RETVAL));
- ST(0) = sv_newmortal() ;
- if (RETVAL == NULL)
- SaveError("%s",dlerror()) ;
- else
- sv_setiv( ST(0), (IV)RETVAL);
-
-
- void
- dl_undef_symbols()
- PPCODE:
-
-
-
- # These functions should not need changing on any platform:
-
- void
- dl_install_xsub(perl_name, symref, filename="$Package")
- char * perl_name
- void * symref
- char * filename
- CODE:
- DLDEBUG(2,fprintf(stderr,"dl_install_xsub(name=%s, symref=%x)\n",
- perl_name, symref));
- ST(0)=sv_2mortal(newRV((SV*)newXS(perl_name, (void(*)())symref, filename)));
-
-
- char *
- dl_error()
- CODE:
- RETVAL = LastError ;
- OUTPUT:
- RETVAL
-
- # end.
-